The damage levels caused by the Palisades Fire are shown across homes built in different decades, with 53.7% of damage assessments categorized as Destroyed (>50%) and 65.9% of homes from the 1970s sustaining No Damage.
Figure 1: The plot is a waffle-style chart that uses pictograms (house icons) to represent the number of homes inspected, categorized by the decade they were built. Each bar corresponds to a decade, with the house icons color-coded to show different damage levels: “No Damage,” “Affected (1-9%),” “Minor (10-25%),” “Major (26-50%),” and “Destroyed (>50%).” This format helps visualize both the quantity of inspections and the severity of damage across time, emphasizing trends like the vulnerability of older homes and the resilience of homes from the 1970s.
How This Graphic Was Made
1. 📦 Load Packages & Setup
Show code
# Load necessary packages using pacman for easier dependency managementpacman::p_load(sf, # For handling shapefiles and geospatial datawaffle, # For creating waffle charts (visualization)showtext, # For adding custom Google fonts and enabling font renderingggtext, # For enhanced text formatting, such as rich text in plotstidyverse, # For comprehensive data manipulation and visualizationglue, # For string interpolation and dynamic text creationcowplot# For combining and customizing plots)# Add local fonts from specified file paths for use in plots and text formattingfont_add("Font Awesome 6", here::here("fonts/otfs/Font Awesome 6 Free-Solid-900.otf"))font_add("Font Awesome 6 Brands", here::here("fonts/otfs/Font Awesome 6 Brands-Regular-400.otf"))font_add(family ="Rockwell-bold", regular ="C:/windows/Fonts/ROCKB.TTF")# Add Google fonts directly, useful for consistent typography across visualsfont_add_google(name ="Roboto", "Roboto")font_add_google(name ="Playfair Display", "Playfair Display")# Enable custom fonts and set rendering options for improved qualityshowtext_auto()# Automatically render custom fonts in graphicsshowtext_opts(dpi =300)# Set font rendering resolution for high-quality output
2. 📖 Load and Prepare Data
Show code
# Load the 'buildings' and 'palisades' datasets using consistent file pathspalisades_path<-here::here("Data/Palisades Inspection")buildings<-st_read(palisades_path)palisades_path<-here::here("Data/Palisades_buildings.gpkg")palisades<-st_read(palisades_path)
3. 🕵 Filter and Transform Data
Show code
# Filter 'palisades' dataset for residential buildings and select relevant columnspalisades<-palisades%>%filter(UseType=="Residential")%>%select(YearBuilt1, HEIGHT, ELEV, geom)# Perform spatial intersection to combine 'buildings' and filtered 'palisades' datadf<-st_intersection(buildings, palisades)# Clean and transform data: handle missing values, filter by year, and categorize decades and damage levelsdf<-df%>%drop_na(YearBuilt1)%>%filter(YearBuilt1>=1916)%>%mutate( decade =case_when(YearBuilt1<1950~"Before 1950",TRUE~glue::glue("{floor(as.numeric(YearBuilt1) / 10) * 10}s")), decade =factor(decade, levels =c("Before 1950", "1950s", "1960s", "1970s", "1980s", "1990s", "2000s", "2010s")), DAMAGE =factor(DAMAGE, levels =c("No Damage", "Affected (1-9%)", "Minor (10-25%)", "Major (26-50%)", "Destroyed (>50%)")))
4. ⌨️ Summarize Data
Show code
# Summarize data: group by DAMAGE and decade, then count occurrencesdamage<-df%>%group_by(DAMAGE, decade)%>%count()%>%ungroup()# Calculate the maximum frequency of "Destroyed (>50%)" as a percentagemax_damage_freq<-df%>%group_by(DAMAGE)%>%count(DAMAGE)%>%mutate(freq =(n/sum(n))*100)%>%slice_max(freq)%>%pull(freq)# Calculate the percentage of homes built before 1960before_1960<-df%>%mutate(homes =case_when(decade%in%c("Before 1950", "1950s")~"before 1960",TRUE~"1960s and rest"))%>%group_by(homes)%>%count(homes)%>%mutate(freq =(n/sum(n)*100))%>%slice(2)%>%pull(freq)# Identify the decade with the highest percentage of "No Damage"best_decade<-df%>%group_by(decade, DAMAGE)%>%count()%>%ungroup()%>%group_by(decade)%>%mutate(freq =n/sum(n)*100)%>%ungroup()%>%filter(DAMAGE%in%"No Damage")%>%slice_max(freq)%>%pull(freq)
5. 🔤 Text
Show code
social<-andresutils::social_caption( bg_color ="#F0F8FF", icon_color ="#3a86ff", font_color ="black", font_family ="Roboto", linkedin ="Andres Gonzalez")title<-toupper("Damage Assessment from the Palisades Fire")st<-paste("One of the wealthiest communities in the U.S. is experiencing the worst fire in its history. ","The Palisades Fire, which began at 10:30 a.m. on January 7, 2025, in Los Angeles County, has caused widespread destruction. ","As of January 18, 2025, out of 3,897 inspected homes, ",paste0("<span style='color:#3a86ff;'>**Destroyed (>50%)**</span> accounted for **", round(max_damage_freq, 1), "%** of all damage assessments. "),paste0("A significant portion of the homes inspected (**", round(before_1960, 1), "%**) were built before 1960, with the majority of these constructed during the 1950s. "),"These older homes, now 66 to over 100 years old, were built across decades and are an integral part of the area's character. ","Other damage levels included <span style='color:#ffbe0b;'>**No Damage**</span>, ","<span style='color:#fb5607;'>**Affected (1-9%)**</span>, ","<span style='color:#ff006e;'>**Minor (10-25%)**</span>, and ","<span style='color:#8338ec;'>**Major (26-50%)**</span>. ","This tragedy highlights the critical need to retrofit older homes with modern fire-resistant materials to mitigate future disasters.")cap<-paste0(st,"<br><br>**Data**: CAL FIRE Damage Inspection (DINS)<br>**Graphic**: ", social)
6. 📊 Plot
Show code
# Create a pictogram chart to visualize damage levels across decadesp<-damage%>%ggplot()+geom_pictogram( mapping =aes( label =DAMAGE, color =DAMAGE, values =n), flip =TRUE, n_rows =20, # Number of rows per pictogram group size =1, # Size of the pictogram icons family ="Font Awesome 6"# Use Font Awesome icons for visualization)+scale_label_pictogram( name =NULL, # No legend title values =rep("home", 5)# Use "home" icon for all damage levels)+scale_colour_manual( values =c("#ffbe0b", "#fb5607", "#ff006e", "#8338ec", "#3a86ff")# Custom color palette for damage levels)+scale_x_discrete( expand =c(0, 0, 0, 0)# No expansion for x-axis)+scale_y_continuous( labels =function(x)format(x*20, big.mark =","), # Format y-axis labels for clarity expand =c(0, 0), # No expansion for y-axis minor_breaks =NULL)+facet_wrap(~decade, nrow =1, strip.position ="bottom")+# Facet by decadeslabs( title =title, # Add title from the earlier narrative subtitle =cap# Add subtitle with detailed text)+coord_fixed()+# Ensure consistent scalingtheme_minimal(# Minimalistic theme for cleaner visuals base_family ="Roboto", # Font family for text base_size =7# Base font size)+theme( legend.position ="none", # Remove legend plot.title.position ="plot", plot.margin =margin(5, 15, 5, 15), # Add padding around the plot plot.background =element_rect(fill ="#F0F8FF", color ="#F0F8FF"), # Light blue background panel.background =element_rect(fill ="#F0F8FF", color ="#F0F8FF"), panel.grid.major =element_line( linewidth =0.3, color ="#CAE9F5"# Light grid lines), plot.title =element_textbox_simple( margin =margin(b =5, t =10), family ="Rockwell-bold", # Font for title size =16), plot.subtitle =element_textbox_simple( margin =margin(b =25, t =10), lineheight =1.2# Adjust subtitle spacing))# Add annotations to emphasize key insights on the plotfinal_plot<-ggdraw(p)+annotate("segment", y =0.44, x =0.65, xend =0.68, color ="#ffbe0b", linewidth =3.2# Highlight a specific area)+geom_richtext(aes( x =0.60, y =0.45, label =paste0("The 1970s had the highest percentage<br>","of **No Damage** inspections at **", round(best_decade, 1), "%**")), size =2, family ="Playfair Display", # Use decorative font fill =NA, label.color =NA# Remove label background and border)+geom_curve(aes(x =0.61, y =0.43, xend =0.515, yend =0.21), arrow =arrow(length =unit(0.1, "cm"), type ="closed"), # Add an arrow for emphasis color ="grey75", linewidth =0.3, curvature =-0.4# Light curve for subtle highlight)
7. 💾 Save
Show code
# Save the final visualization as a PNG fileggsave("Palisades Houses.png", plot =final_plot, height =5, width =8)
8. 🚀 GitHub Repository
Expand for GitHub Repo
Explore the complete code for this visualization in the following Quarto file: Palisades Houses.qmd.
For additional visualizations and projects, click here.
Do you enjoy my blog? Subscribe here to get notifications and updates (it's free!):
Source Code
---title: "Damage Assessment From The Palisades Fire"description: "The damage levels caused by the Palisades Fire are shown across homes built in different decades, with 53.7% of damage assessments categorized as Destroyed (>50%) and 65.9% of homes from the 1970s sustaining No Damage."lightbox: truedate: January 20, 2025categories: [R Programming, Data Visualization, Waffle]tags: [ggplot2, data-visualization, tidyverse, Waffle]image: "Palisades Houses.png"editor: markdown: wrap: sentence---{#fig-1}### <mark> **How This Graphic Was Made** </mark>#### 1. 📦 Load Packages & Setup```{r}#| label: load#| warning: false#| message: false#| results: hide# Load necessary packages using pacman for easier dependency managementpacman::p_load( sf, # For handling shapefiles and geospatial data waffle, # For creating waffle charts (visualization) showtext, # For adding custom Google fonts and enabling font rendering ggtext, # For enhanced text formatting, such as rich text in plots tidyverse, # For comprehensive data manipulation and visualization glue, # For string interpolation and dynamic text creation cowplot # For combining and customizing plots)# Add local fonts from specified file paths for use in plots and text formattingfont_add("Font Awesome 6", here::here("fonts/otfs/Font Awesome 6 Free-Solid-900.otf"))font_add("Font Awesome 6 Brands", here::here("fonts/otfs/Font Awesome 6 Brands-Regular-400.otf"))font_add(family ="Rockwell-bold", regular ="C:/windows/Fonts/ROCKB.TTF")# Add Google fonts directly, useful for consistent typography across visualsfont_add_google(name ="Roboto", "Roboto")font_add_google(name ="Playfair Display", "Playfair Display")# Enable custom fonts and set rendering options for improved qualityshowtext_auto() # Automatically render custom fonts in graphicsshowtext_opts(dpi =300) # Set font rendering resolution for high-quality output```#### 2. 📖 Load and Prepare Data```{r}#| label: read#| include: true#| eval: true#| warning: false#| results: hide# Load the 'buildings' and 'palisades' datasets using consistent file pathspalisades_path <- here::here("Data/Palisades Inspection")buildings <-st_read(palisades_path)palisades_path <- here::here("Data/Palisades_buildings.gpkg")palisades <-st_read(palisades_path)```#### 3. 🕵 Filter and Transform Data```{r}#| label: filter and transform#| include: true#| eval: true#| results: hide#| warning: false# Filter 'palisades' dataset for residential buildings and select relevant columnspalisades <- palisades %>%filter(UseType =="Residential") %>%select(YearBuilt1, HEIGHT, ELEV, geom)# Perform spatial intersection to combine 'buildings' and filtered 'palisades' datadf <-st_intersection(buildings, palisades)# Clean and transform data: handle missing values, filter by year, and categorize decades and damage levelsdf <- df %>%drop_na(YearBuilt1) %>%filter(YearBuilt1 >=1916) %>%mutate(decade =case_when( YearBuilt1 <1950~"Before 1950",TRUE~ glue::glue("{floor(as.numeric(YearBuilt1) / 10) * 10}s") ),decade =factor(decade, levels =c("Before 1950", "1950s", "1960s", "1970s", "1980s", "1990s", "2000s", "2010s")),DAMAGE =factor(DAMAGE, levels =c("No Damage", "Affected (1-9%)", "Minor (10-25%)", "Major (26-50%)", "Destroyed (>50%)")) )```#### 4. ⌨️ Summarize Data```{r}#| label: summarize#| warning: false#| results: hide# Summarize data: group by DAMAGE and decade, then count occurrencesdamage <- df %>%group_by(DAMAGE, decade) %>%count() %>%ungroup()# Calculate the maximum frequency of "Destroyed (>50%)" as a percentagemax_damage_freq <- df %>%group_by(DAMAGE) %>%count(DAMAGE) %>%mutate(freq = (n /sum(n)) *100) %>%slice_max(freq) %>%pull(freq)# Calculate the percentage of homes built before 1960before_1960 <- df %>%mutate(homes =case_when( decade %in%c("Before 1950", "1950s") ~"before 1960",TRUE~"1960s and rest" )) %>%group_by(homes) %>%count(homes) %>%mutate(freq = (n /sum(n) *100)) %>%slice(2) %>%pull(freq)# Identify the decade with the highest percentage of "No Damage"best_decade <- df %>%group_by(decade, DAMAGE) %>%count() %>%ungroup() %>%group_by(decade) %>%mutate(freq = n /sum(n) *100) %>%ungroup() %>%filter(DAMAGE %in%"No Damage") %>%slice_max(freq) %>%pull(freq)```#### 5. 🔤 Text```{r}#| label: text#| include: true#| warning: falsesocial <- andresutils::social_caption(bg_color ="#F0F8FF",icon_color ="#3a86ff",font_color ="black",font_family ="Roboto",linkedin ="Andres Gonzalez")title <-toupper("Damage Assessment from the Palisades Fire")st <-paste("One of the wealthiest communities in the U.S. is experiencing the worst fire in its history. ","The Palisades Fire, which began at 10:30 a.m. on January 7, 2025, in Los Angeles County, has caused widespread destruction. ","As of January 18, 2025, out of 3,897 inspected homes, ",paste0("<span style='color:#3a86ff;'>**Destroyed (>50%)**</span> accounted for **", round(max_damage_freq, 1), "%** of all damage assessments. "),paste0("A significant portion of the homes inspected (**", round(before_1960, 1), "%**) were built before 1960, with the majority of these constructed during the 1950s. "),"These older homes, now 66 to over 100 years old, were built across decades and are an integral part of the area's character. ","Other damage levels included <span style='color:#ffbe0b;'>**No Damage**</span>, ","<span style='color:#fb5607;'>**Affected (1-9%)**</span>, ","<span style='color:#ff006e;'>**Minor (10-25%)**</span>, and ","<span style='color:#8338ec;'>**Major (26-50%)**</span>. ","This tragedy highlights the critical need to retrofit older homes with modern fire-resistant materials to mitigate future disasters.")cap <-paste0( st,"<br><br>**Data**: CAL FIRE Damage Inspection (DINS)<br>**Graphic**: ", social)```#### 6. 📊 Plot```{r}#| label: plot#| warning: false# Create a pictogram chart to visualize damage levels across decadesp <- damage %>%ggplot() +geom_pictogram(mapping =aes(label = DAMAGE, color = DAMAGE, values = n ),flip =TRUE,n_rows =20, # Number of rows per pictogram groupsize =1, # Size of the pictogram iconsfamily ="Font Awesome 6"# Use Font Awesome icons for visualization ) +scale_label_pictogram(name =NULL, # No legend titlevalues =rep("home", 5) # Use "home" icon for all damage levels ) +scale_colour_manual(values =c("#ffbe0b", "#fb5607", "#ff006e", "#8338ec", "#3a86ff") # Custom color palette for damage levels ) +scale_x_discrete(expand =c(0, 0, 0, 0) # No expansion for x-axis ) +scale_y_continuous(labels =function(x) format(x *20, big.mark =","), # Format y-axis labels for clarityexpand =c(0, 0), # No expansion for y-axisminor_breaks =NULL ) +facet_wrap(~decade, nrow =1, strip.position ="bottom") +# Facet by decadeslabs(title = title, # Add title from the earlier narrativesubtitle = cap # Add subtitle with detailed text ) +coord_fixed() +# Ensure consistent scalingtheme_minimal( # Minimalistic theme for cleaner visualsbase_family ="Roboto", # Font family for textbase_size =7# Base font size ) +theme(legend.position ="none", # Remove legendplot.title.position ="plot",plot.margin =margin(5, 15, 5, 15), # Add padding around the plotplot.background =element_rect(fill ="#F0F8FF", color ="#F0F8FF"), # Light blue backgroundpanel.background =element_rect(fill ="#F0F8FF", color ="#F0F8FF"),panel.grid.major =element_line(linewidth =0.3,color ="#CAE9F5"# Light grid lines ),plot.title =element_textbox_simple(margin =margin(b =5, t =10),family ="Rockwell-bold", # Font for titlesize =16 ),plot.subtitle =element_textbox_simple(margin =margin(b =25, t =10),lineheight =1.2# Adjust subtitle spacing ) )# Add annotations to emphasize key insights on the plotfinal_plot <-ggdraw(p) +annotate("segment", y =0.44, x =0.65, xend =0.68, color ="#ffbe0b", linewidth =3.2# Highlight a specific area ) +geom_richtext(aes(x =0.60, y =0.45, label =paste0("The 1970s had the highest percentage<br>","of **No Damage** inspections at **", round(best_decade, 1), "%**" ) ),size =2, family ="Playfair Display", # Use decorative fontfill =NA, label.color =NA# Remove label background and border ) +geom_curve(aes(x =0.61, y =0.43, xend =0.515, yend =0.21), arrow =arrow(length =unit(0.1, "cm"), type ="closed"), # Add an arrow for emphasiscolor ="grey75", linewidth =0.3, curvature =-0.4# Light curve for subtle highlight )```#### 7. 💾 Save```{r}#| label: save#| warning: false# Save the final visualization as a PNG fileggsave("Palisades Houses.png", plot = final_plot, height =5, width =8)```#### 8. 🚀 GitHub Repository::: {.callout-tip collapse="true"}##### Expand for GitHub RepoExplore the complete code for this visualization in the following Quarto file: [`Palisades Houses.qmd`](https://github.com/OKcomputer626/Andres-Portfolio/blob/master/visualization/Visualizations/2025/LA%20Wildfire/Palisades%20Houses.qmd).For additional visualizations and projects, [click here](https://github.com/OKcomputer626/Andres-Portfolio).:::